<!DOCTYPE html>
<html>
<head>
<title>第7章：Ajax</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<style type="text/css">
html,
body,
h1,
h2,
h3,
h4,
h5,
h6,
p,
blockquote,
ol,
ul,
li,
img {
    margin: 0;
    padding: 0;
    font-size: 100%;
    font: inherit;
    vertical-align: baseline;
}

html * {
    font-family: "ff-din-web-pro-1", "ff-din-web-pro-2", sans-serif;
    font-size: 16px;
    line-height: 19.2px;
    color-profile: sRGB;
}

body {
    min-width: 32em;
    max-width: 56em;
    margin: 10px auto;
}

p, blockquote p {
    line-height: 1.6;
}

ul, ol {
    margin: 16px 0;
}

ul li, ol li {
    line-height: 1.6;
}

p {
    font-weight: lighter;
    margin: 10px 0;
}

strong {
    font-weight: bold;
}

ol,
ul {
    margin-left: 2em;
}

h1,
h2,
h3,
h4,
h5,
h6 {
    font-weight: lighter;
    text-transform: capitalize;
    margin: 20px 0;
    border-bottom: 1px solid;
    padding-bottom: 6px;
}

h1, h1 > code {
    font-size: 24.624px;
    line-height: 29.548799999999996px;
}

h2, h2 > code {
    font-size: 24.624px;
    line-height: 29.548799999999996px;
}

h3, h3 > code {
    font-size: 23.44px;
    line-height: 28.128px;
}

h4, h4 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

h5, h5 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

h6, h6 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

img {
    margin-bottom: 20px;
}

h1 img,
h2 img,
h3 img,
h4 img,
h5 img,
h6 img,
p img {
    margin-bottom: 0;
}

pre,
code {
    font-family: monospace, Consolas, "Source Code Pro", Arial, sans-serif;
    color: #586e75;
    background-color: #eee8d5;
}

pre {
    white-space: pre-wrap;
    word-wrap: break-word;
    padding: 12px;
    margin-bottom: 20px;
}

code {
    border-radius: 3px;
}

h1 {
    text-transform: uppercase;
    font-weight: bold;
}

h3,
h4,
h5,
h6 {
    border-bottom: none;
}

html body {
    background-color: #fdf6e3;
}

html h1,
html h2,
html h3,
html h4,
html h5,
html h6 {
    color: #586e75;
    border-color: #657b83;
}

html a,
html a:active,
html a:visited {
    color: #586e75;
    text-decoration: none;
    border-bottom: 1px dashed;
    border-radius: 2px;
}

html a:hover {
    background-color: #eee8d5;
}

blockquote a:hover {
    background-color: #fdf6e3;
}

html a,
html a:active,
html a:visited,
html code.url {
    color: #b58900;
}

html h1 {
    color: #b58900;
}

html h2,
html h3,
html h4,
html h5,
html h6 {
    color: #b58900;
}

/* QUOTES
=============================================================================*/
blockquote {
    border-left: 4px solid #b58900;
    padding: 12px;
    background: #eee8d5;
    border-bottom-right-radius: 2px;
}

blockquote code {
    background: #fdf6e3;
}

blockquote > :first-child {
    margin-top: 0;
}

blockquote > :last-child {
    margin-bottom: 0;
}

/* TABLES
=============================================================================*/
table {
    margin: 0 auto;
    border-collapse: collapse;
    width: 100%;
    box-sizing: border-box;
    margin-bottom: 30px;
}

table th, table td {
    border: 1px solid #ccc;
    padding: 6px 13px;
}

table td {
    word-break: break-word;
    line-height: 1.3;
}

table th {
    font-weight: bold;
    text-align: center !important;
    background-color: #eee8d5;
}

table tr {
    border-top: 1px solid #ccc;
    background-color: #fdf6e3;
}

/* IMAGES
=============================================================================*/
img {
    max-width: 100%;
}

p > img {
    display: table;
    margin: 0 auto;
}

p code, li code, td code {
    padding: 1px 3px;
    border-radius: 3px;
}

.cp_embed_wrapper {
    margin: 20px 0;
}

.hljs {
	background: #eee8d5 !important;
}

@media screen and (min-width: 980px) and (max-width: 980px) {	
    table thead tr th,
    table thead tr th > code,
    table tbody tr td,
    table tbody tr td > code,
    table tbody tr td > strong {
        font-size: 1.3em;
        line-height: 1.3;
    }

    p, p code,
    p strong, p strong > code,
    blockquote p {
        font-size: 1.3em;
        line-height: 1.6;
    }

    pre > code,
    ul li pre > code,
    ol li pre > code{
		font-size: 1.3em;
        line-height: 1.3;    	
    }	
	

    ul li, ol li,
    ul li > code,
    ol li > code {
        font-size: 1.3em;
		line-height: 1.3;          
    }

    ul {
        margin-left: 3.4em;
    }

    ol {
        margin-left: 3.6em;
    }
}
</style>
<style type="text/css">
.highlight  { background: #ffffff; }
.highlight .c { color: #999988; font-style: italic } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { font-weight: bold } /* Keyword */
.highlight .o { font-weight: bold } /* Operator */
.highlight .cm { color: #999988; font-style: italic } /* Comment.Multiline */
.highlight .cp { color: #999999; font-weight: bold } /* Comment.Preproc */
.highlight .c1 { color: #999988; font-style: italic } /* Comment.Single */
.highlight .cs { color: #999999; font-weight: bold; font-style: italic } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .gd .x { color: #000000; background-color: #ffaaaa } /* Generic.Deleted.Specific */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #999999 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .gi .x { color: #000000; background-color: #aaffaa } /* Generic.Inserted.Specific */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #aaaaaa } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { font-weight: bold } /* Keyword.Constant */
.highlight .kd { font-weight: bold } /* Keyword.Declaration */
.highlight .kp { font-weight: bold } /* Keyword.Pseudo */
.highlight .kr { font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #445588; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #009999 } /* Literal.Number */
.highlight .s { color: #d14 } /* Literal.String */
.highlight .na { color: #008080 } /* Name.Attribute */
.highlight .nb { color: #0086B3 } /* Name.Builtin */
.highlight .nc { color: #445588; font-weight: bold } /* Name.Class */
.highlight .no { color: #008080 } /* Name.Constant */
.highlight .ni { color: #800080 } /* Name.Entity */
.highlight .ne { color: #990000; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #990000; font-weight: bold } /* Name.Function */
.highlight .nn { color: #555555 } /* Name.Namespace */
.highlight .nt { color: #000080 } /* Name.Tag */
.highlight .nv { color: #008080 } /* Name.Variable */
.highlight .ow { font-weight: bold } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mf { color: #009999 } /* Literal.Number.Float */
.highlight .mh { color: #009999 } /* Literal.Number.Hex */
.highlight .mi { color: #009999 } /* Literal.Number.Integer */
.highlight .mo { color: #009999 } /* Literal.Number.Oct */
.highlight .sb { color: #d14 } /* Literal.String.Backtick */
.highlight .sc { color: #d14 } /* Literal.String.Char */
.highlight .sd { color: #d14 } /* Literal.String.Doc */
.highlight .s2 { color: #d14 } /* Literal.String.Double */
.highlight .se { color: #d14 } /* Literal.String.Escape */
.highlight .sh { color: #d14 } /* Literal.String.Heredoc */
.highlight .si { color: #d14 } /* Literal.String.Interpol */
.highlight .sx { color: #d14 } /* Literal.String.Other */
.highlight .sr { color: #009926 } /* Literal.String.Regex */
.highlight .s1 { color: #d14 } /* Literal.String.Single */
.highlight .ss { color: #990073 } /* Literal.String.Symbol */
.highlight .bp { color: #999999 } /* Name.Builtin.Pseudo */
.highlight .vc { color: #008080 } /* Name.Variable.Class */
.highlight .vg { color: #008080 } /* Name.Variable.Global */
.highlight .vi { color: #008080 } /* Name.Variable.Instance */
.highlight .il { color: #009999 } /* Literal.Number.Integer.Long */
.pl-c {
    color: #969896;
}

.pl-c1,.pl-mdh,.pl-mm,.pl-mp,.pl-mr,.pl-s1 .pl-v,.pl-s3,.pl-sc,.pl-sv {
    color: #0086b3;
}

.pl-e,.pl-en {
    color: #795da3;
}

.pl-s1 .pl-s2,.pl-smi,.pl-smp,.pl-stj,.pl-vo,.pl-vpf {
    color: #333;
}

.pl-ent {
    color: #63a35c;
}

.pl-k,.pl-s,.pl-st {
    color: #a71d5d;
}

.pl-pds,.pl-s1,.pl-s1 .pl-pse .pl-s2,.pl-sr,.pl-sr .pl-cce,.pl-sr .pl-sra,.pl-sr .pl-sre,.pl-src,.pl-v {
    color: #df5000;
}

.pl-id {
    color: #b52a1d;
}

.pl-ii {
    background-color: #b52a1d;
    color: #f8f8f8;
}

.pl-sr .pl-cce {
    color: #63a35c;
    font-weight: bold;
}

.pl-ml {
    color: #693a17;
}

.pl-mh,.pl-mh .pl-en,.pl-ms {
    color: #1d3e81;
    font-weight: bold;
}

.pl-mq {
    color: #008080;
}

.pl-mi {
    color: #333;
    font-style: italic;
}

.pl-mb {
    color: #333;
    font-weight: bold;
}

.pl-md,.pl-mdhf {
    background-color: #ffecec;
    color: #bd2c00;
}

.pl-mdht,.pl-mi1 {
    background-color: #eaffea;
    color: #55a532;
}

.pl-mdr {
    color: #795da3;
    font-weight: bold;
}

.pl-mo {
    color: #1d3e81;
}
.task-list {
padding-left:10px;
margin-bottom:0;
}

.task-list li {
    margin-left: 20px;
}

.task-list-item {
list-style-type:none;
padding-left:10px;
}

.task-list-item label {
font-weight:400;
}

.task-list-item.enabled label {
cursor:pointer;
}

.task-list-item+.task-list-item {
margin-top:3px;
}

.task-list-item-checkbox {
display:inline-block;
margin-left:-20px;
margin-right:3px;
vertical-align:1px;
}
</style>
<base target=_blank>
<meta http-equiv="X-UA-Compatible" content="IE=edge,Chrome=1">
<meta name="keywords" content="whjin,前端开发文档,html,css,javascript,canvas,jquery,vue.js,http,ajax,git,webpack">
<meta name="format-detection" content="telephone=no">
<meta name="description" content="前端开发文档">
<meta name="author" content="whjin">
<link rel="shortcut icon" href="https://whjin.github.io/frontend-dev-doc/images/logo.png">
<a href="https://github.com/whjin" class="github-corner" aria-label="View source on GitHub" target="_blank"><svg width="80" height="80" viewBox="0 0 250 250" style="fill:#151513; color:#fff; position: fixed; top: 0; border: 0; right: 0;" aria-hidden="true"><path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path><path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" fill="currentColor" style="transform-origin: 130px 106px;" class="octo-arm"></path><path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" fill="currentColor" class="octo-body"></path></svg></a><style>.github-corner:hover .octo-arm{animation:octocat-wave 560ms ease-in-out}@keyframes octocat-wave{0%,100%{transform:rotate(0)}20%,60%{transform:rotate(-25deg)}40%,80%{transform:rotate(10deg)}}@media (max-width:500px){.github-corner:hover .octo-arm{animation:none}.github-corner .octo-arm{animation:octocat-wave 560ms ease-in-out}}</style>
<link href="https://cdn.bootcss.com/highlight.js/9.15.6/styles/a11y-light.min.css" rel="stylesheet">
<script src="https://cdn.bootcss.com/highlight.js/9.15.6/highlight.min.js"></script>
<script >hljs.initHighlightingOnLoad();</script> 
</head>
<body>
<h1 id="-7-ajax">第7章：Ajax</h1>
<p>Ajax是高性能JS的基础，它可以通过延迟下载体积较大的资源文件来使得页面加载速度更快。它通过异步的方式在客户端和服务端之间传输数据。它甚至可以只用一个HTTP请求就获取整个页面的资源。</p>
<h2 id="-">数据传输</h2>
<p>Ajax是一种与服务器通信而无需重载页面的方法。</p>
<h3 id="-">请求数据</h3>
<p>有五种常用技术用于向服务器请求数据：</p>
<ol>
<li><code>XMLHTTPRequest</code></li><li>动态脚本注入</li><li><code>iframes</code></li><li><code>Comet</code></li><li>Multipart XHR</li></ol>
<p>JS中经常使用的三种技术：XHR、动态脚本注入和Multipart XHR。</p>
<p><strong>XMLHTTPRequest</strong></p>
<p><code>XMLHTTPRequest</code>（XHR）是目前最常用的技术，它允许异步发送和接收数据。所有主流浏览器都对他提供了支持，而且它还能精确地控制发送请求和接收数据。</p>
<p><strong>使用XHR，POST和GET的对比。</strong>只有当请求的URL加上参数的长度接近或超过<strong>2048</strong>个字符时，才应该用POST请求获取数据。</p>
<p><strong>动态脚本注入</strong></p>
<p>这种技术克服了XHR的最大限制：它能跨域请求数据。使用JS创建新的脚本标签，并设置它的<code>src</code>属性为不同域的URL。</p>
<pre><code>var script = document.createElement(&#39;script&#39;);
script.src = &#39;http://domain.com/js/lib.js&#39;;
document.getElementsByTagName(&#39;head&#39;)[0].appendChild(script);
</code></pre><p>动态脚本注入不能设置请求的头信息，只能使用GET方式，不能设置请求的超时处理或重试。特别重要的一点是，因为响应消息作为脚本标签的源码，它必须是可执行的JS代码。其他的任何格式都必须封装在一个回调函数中。</p>
<pre><code>function jsonCallback(jsonString) {
    var data = eval(&#39;(&#39; + jsonString + &#39;)&#39;);
    //处理数据
}
</code></pre><p><strong>Multipart XHR</strong></p>
<p>Multipart XHR允许客户端只用一个HTTP请求就可以从服务端向客户端传输多个资源。它通过在服务端将资源打包成一个有双方约定的字符串分割的长字符串并发送给客户端。然后用JS代码处理这个长字符串，并根据它的<code>mime-type</code>类型和传入的其他“头信息”解析出每个资源。</p>
<p><p data-height="300" data-theme-id="0" data-slug-hash="OwPWLG" data-default-tab="js" data-user="whjin" data-embed-version="2" data-pen-title="MXHR" class="codepen">See the Pen <a href="https://codepen.io/whjin/pen/OwPWLG/">MXHR</a> by whjin (<a href="https://codepen.io/whjin">@whjin</a>) on <a href="https://codepen.io">CodePen</a>.</p></p>
<script async src="https://static.codepen.io/assets/embed/ei.js"></script>

<p>使用这个技术最大缺点是获得的资源不能被浏览器缓存。合并后的资源是作为字符串传输，然后被JS代码分解成片段，无法用编程的方式向浏览器缓存里注入文件，因此用这种方式获取的资源无法被缓存。</p>
<p>MXHR能显著提升页面的整体性能：</p>
<ul>
<li>页面包含了大量其他地方用不到的资源（无需缓存），尤其是图片</li><li>网站已经在每个页面使用一个独立打包的JS或CSS文件来减少HTTP请求；因此对每个页面来说这些文件都是唯一的。所以不需要从缓存中读取数据，除非重载页面</li></ul>
<h3 id="-">发送数据</h3>
<p>当数据只需发送到服务器时，有两种广泛使用的技术：XHR和信标。</p>
<p><strong>XMLHTTPRequest</strong></p>
<p>XHR既可以用于从服务器获取数据，也可以用于把数据传回服务器。当需要传回的数据超出浏览器的最大URL尺寸限制时，XHR特别有用。</p>
<p><p data-height="265" data-theme-id="0" data-slug-hash="OwPWxJ" data-default-tab="js" data-user="whjin" data-embed-version="2" data-pen-title="XHR" class="codepen">See the Pen <a href="https://codepen.io/whjin/pen/OwPWxJ/">XHR</a> by whjin (<a href="https://codepen.io/whjin">@whjin</a>) on <a href="https://codepen.io">CodePen</a>.</p></p>
<script async src="https://static.codepen.io/assets/embed/ei.js"></script>

<p><strong>Beacons</strong></p>
<p>这个技术非常类似动态脚本注入。使用JS创建一个新的<code>Image</code>对象，并把<code>src</code>属性设置为服务器上脚本的URL。该URL包含了要通过GET传回的键值对数据。</p>
<pre><code>var url = &#39;/status_tracker.php&#39;;
var params = [&#39;step=2&#39;, &#39;time=12472112&#39;];
(new Image()).src = url + &#39;?&#39; + params.join(&#39;&amp;&#39;);
</code></pre><p>服务器会接收到数据并保存下来，无须向客户端发送任何反馈信息。它的性能消耗很小，而且服务端的错误完全不会影响到客户端。</p>
<p>图片信标无法发送POST请求，而URL的长度有最大值，可以发送的数据长度被限制的相当小。可以接收服务器返回的数据，但只局限于非常少的几种方式。一种是<code>Image</code>对象的<code>load</code>事件，它会告知服务器是否成功接收数据。还可以检查服务器返回的图片的宽高，并使用这些数字通知服务器的状态。</p>
<p>信标是向服务器回传数据最快且最有效的方式。服务器根本不需要发送任何响应正文，因此无需担心客户端下载数据。唯一的缺点是能够接收到的响应类型是有限的。如果需要返回大量数据给客户端，可以使用XHR。如果只关心发送数据到服务器，可以使用图片信标。</p>
<h2 id="-">数据格式</h2>
<p>当使用数据传输技术时，必须考虑这些因素：功能集、兼容性、性能以及数据传输方向。当考虑数据格式时，唯一需要比较的标准就是速度。</p>
<h3 id="xmlp134">XMLP134</h3>
<h3 id="xpath">XPath</h3>
<p><strong>响应内容大小和解析事件P137</strong></p>
<p>XML和XPath两项技术略过不讲。</p>
<h3 id="json">JSON</h3>
<p>在JS中可以简单地使用<code>eval()</code>来解析JSON字符串。</p>
<pre><code>function parseJSON(responseText) {
    return eval(&#39;(&#39; + responseText + &#39;)&#39;)
}
</code></pre><blockquote>
<p>在代码中使用<code>eval()</code>是很危险的，特别是它执行第三方的JSON数据。尽可能使用<code>JSON.parse()</code>方法解析字符串本身。该方法可以捕获JSON中的词法错误，并允许传入一个函数，用来过滤或转换解析结果。</p>
</blockquote>
<h3 id="json-p">JSON-P</h3>
<p>JSONP在本地执行会导致性能问题。当使用XHR时，JSON数据被当成字符串返回。该字符串紧接着被<code>eval()</code>转换成原生对象。然而在使用动态脚本注入时，JSON数据被当成另一个JS文件并作为原生代码执行。为实现这一点，这些数据必须封装在一个回调函数中。</p>
<pre><code>parseJSON([
    {&quot;id&quot;: 1, &quot;username&quot;: &quot;Andy&quot;, &quot;realname&quot;: &quot;Andy Liu&quot;, &quot;email&quot;: &quot;AndyLiu@163.com&quot;},
    {&quot;id&quot;: 2, &quot;username&quot;: &quot;Lucy&quot;, &quot;realname&quot;: &quot;Lucy Chen&quot;, &quot;email&quot;: &quot;LucyChen@163.com&quot;},
]);
</code></pre><p>由于数据是当做原生的JS，因此解析速度跟原生JS一样快。最快的JSON格式是使用数组形式的JSON-P。</p>
<blockquote>
<p>有一种情形要避免使用JSON-P（与性能无关），因为JSON-P必须是可执行的JS，它可能被任何人调用并使用动态脚本注入技术插入到任何网站。不要把任何敏感数据编码在JSON-P中，因为无法确认它是否保持着私有调用状态。</p>
</blockquote>
<h3 id="html">HTML</h3>
<p>HTML作为一种数据格式，既缓慢，有臃肿。</p>
<h3 id="-">自定义格式</h3>
<p>控制字符在JS中使用Unicode表示法，<code>split()</code>函数可以使用字符串或正则表达式作为参数。如果在数据中存在空白字段，使用字符串作为分隔符；如果分隔符是正则表达式，IE中的<code>split()</code>会忽略紧挨着的两个分隔符中的第二个分隔符。</p>
<pre><code>//正则表达式作为分隔符
var rows = req.responseText.split(/\u0001/);

//字符串作为分隔符
var rows = req.responseText.split(&quot;\u0001&quot;);
</code></pre><h3 id="-">数据格式总结</h3>
<p>如果是数据集很大且对解析时间有要求，那么从以下两种格式中做出选择：</p>
<ol>
<li>JSON-P数据，使用动态脚本注入获取。它把数据当做可执行JS而不是字符串，解析速度极快。它能跨域使用，但涉及敏感数据时不应该使用它。</li><li>字符分隔的自定义格式，使用XHR或动态脚本注入获取，使用<code>split()</code>解析。这项技术解析大数据集比JSON-P略快，而且通常文件尺寸更小。</li></ol>
<h2 id="ajax-">Ajax性能指南</h2>
<h3 id="-">缓存数据</h3>
<p>最快的AJax请求就是没有请求。有两种主要的方法可避免发送不必要的请求：</p>
<ol>
<li>在服务端，设置HTTP头信息以确保响应会被浏览器缓存。</li><li>在客户端，把获取到的信息存储到本地，从而避免再次请求。</li></ol>
<h4 id="-http-">设置HTTP头信息</h4>
<p>如果希望Ajax响应能够被浏览器缓存，必须使用GET方式发出请求，还必须在响应中发送正确的HTTP头信息。Expires头信息会告诉浏览器应该缓存响应多久。它的值是一个日期，过期后对该URL的任何请求都不再从缓存中获取，而是重新访问服务器。</p>
<p>设置Expires头信息是确保浏览器缓存Ajax响应最简单的方法。</p>
<h4 id="-">本地数据存储</h4>
<p><p data-height="300" data-theme-id="0" data-slug-hash="MBYreG" data-default-tab="js" data-user="whjin" data-embed-version="2" data-pen-title="xhrLocalCache" class="codepen">See the Pen <a href="https://codepen.io/whjin/pen/MBYreG/">xhrLocalCache</a> by whjin (<a href="https://codepen.io/whjin">@whjin</a>) on <a href="https://codepen.io">CodePen</a>.</p></p>
<script async src="https://static.codepen.io/assets/embed/ei.js"></script>

<h2 id="-ajax-p148">了解Ajax类库的局限P148</h2>
<h2 id="-">小结</h2>
<p>动态脚本注入允许跨域请求和本地执行JS和JSON，但是它的接口不是很安全，而且不能读取头信息或响应代码。MXHR可以用来减少请求数，并处理一个响应中的各种文件类型，但是它不能缓存接收到的响应。当需要发送数据时，图片信标是一种简单而有效的方法。XHR还可以用POST发送大量数据。</p>
<p>除了这些格式和传输技术，还有一些准则有助于加速Ajax：</p>
<ul>
<li>减少请求数，可通过合并JS和CSS文件，或使用MXHR。</li><li>缩短页面的加载时间，页面主要内容加载完成后，用Ajax获取那些次要的文件。</li><li>确保代码错误不会输出给用户，并在服务端处理错误。</li><li>知道何时使用成熟的Ajax类库，以及何时编写自己的底层AJax代码。</li></ul>

</body>
</html>
<!-- This document was created with MarkdownPad, the Markdown editor for Windows (http://markdownpad.com) -->
